2016-02-22 11:00:01 -05:00
|
|
|
import lldb
|
2020-01-16 16:38:04 +00:00
|
|
|
import traceback
|
2016-02-22 11:00:01 -05:00
|
|
|
|
|
|
|
def print_frames(thread, num_frames, current_thread):
|
|
|
|
# TODO: Make output header similar to bt.
|
2020-01-16 16:38:04 +00:00
|
|
|
print('%c thread #%i' % ('*' if current_thread else ' ', thread.idx))
|
2016-02-22 11:00:01 -05:00
|
|
|
|
|
|
|
if current_thread:
|
|
|
|
selected_frame = thread.GetSelectedFrame()
|
|
|
|
|
|
|
|
for frame in thread.frames[:+num_frames]:
|
|
|
|
pc = str(frame.addr)
|
|
|
|
var = frame
|
2017-06-07 13:16:24 +00:00
|
|
|
function_name = frame.GetFunctionName()
|
2018-01-24 17:04:36 +00:00
|
|
|
if function_name == "interp_exec_method_full":
|
2017-06-07 13:16:24 +00:00
|
|
|
try:
|
2018-01-24 17:04:36 +00:00
|
|
|
s = 'frame->imethod->method'
|
2020-01-16 16:38:04 +00:00
|
|
|
methoddesc = frame.EvaluateExpression('(char*) mono_method_full_name (' + s + ', 1)').summary[1:-1]
|
2017-06-07 13:16:24 +00:00
|
|
|
|
|
|
|
ipoffset = frame.EvaluateExpression('ip').GetValueAsUnsigned()
|
|
|
|
insn = ''
|
|
|
|
if ipoffset != 0:
|
2020-01-16 16:38:04 +00:00
|
|
|
ipoffset -= frame.EvaluateExpression('frame->imethod->code').GetValueAsUnsigned()
|
|
|
|
insn = "\"" + frame.EvaluateExpression('mono_interp_opname (*ip)').summary[1:-1] + "\""
|
|
|
|
var = '%s @ %d %s || %s' % (methoddesc, ipoffset, insn, frame)
|
2017-06-07 13:16:24 +00:00
|
|
|
except Exception as e:
|
2020-01-16 16:38:04 +00:00
|
|
|
traceback.print_exc()
|
2017-06-07 13:16:24 +00:00
|
|
|
elif function_name == "mono_interp_transform_method":
|
|
|
|
try:
|
|
|
|
s = 'runtime_method->method'
|
|
|
|
klassname = frame.EvaluateExpression('(char*) ' + s + '->klass->name').summary[1:-1]
|
|
|
|
methodname = frame.EvaluateExpression('(char*) ' + s + '->name').summary[1:-1]
|
|
|
|
var = 'transforming %s::%s || %s' % (klassname, methodname, frame)
|
|
|
|
except Exception as e:
|
2020-01-16 16:38:04 +00:00
|
|
|
print("DBG: transformfail:" + str(e))
|
2017-06-07 13:16:24 +00:00
|
|
|
elif pc[0] == '0':
|
2016-02-22 11:00:01 -05:00
|
|
|
try:
|
|
|
|
framestr = frame.EvaluateExpression('(char*)mono_pmip((void*)%s)' % pc).summary[1:-1]
|
|
|
|
var = 'frame #%i: %s%s' % (frame.idx, pc, framestr)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
2020-01-16 16:38:04 +00:00
|
|
|
print(' %c %s' % ('*' if current_thread and frame.idx == selected_frame.idx else ' ', var))
|
2016-02-22 11:00:01 -05:00
|
|
|
|
|
|
|
def monobt(debugger, command, result, dict):
|
|
|
|
opts = {'all_bt': False, 'num_frames': None}
|
|
|
|
|
|
|
|
if command == 'all':
|
|
|
|
opts['all_bt'] = True
|
|
|
|
elif command.isdigit():
|
|
|
|
opts['num_frames'] = int(command)
|
|
|
|
elif command != '':
|
2020-01-16 16:38:04 +00:00
|
|
|
print('error: monobt [<number>|all]')
|
2016-02-22 11:00:01 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
target = debugger.GetSelectedTarget()
|
|
|
|
process = target.process
|
|
|
|
|
|
|
|
if not process.IsValid():
|
2020-01-16 16:38:04 +00:00
|
|
|
print('error: invalid process')
|
2016-02-22 11:00:01 -05:00
|
|
|
return
|
|
|
|
|
|
|
|
if opts['all_bt']:
|
|
|
|
for thread in process.threads:
|
|
|
|
print_frames(thread, len(thread), process.selected_thread == thread)
|
2020-01-16 16:38:04 +00:00
|
|
|
print('')
|
2016-02-22 11:00:01 -05:00
|
|
|
else:
|
|
|
|
thread = process.selected_thread
|
|
|
|
num_frames = len(thread) if opts['num_frames'] is None else opts['num_frames']
|
|
|
|
print_frames(thread, num_frames, True)
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
def __lldb_init_module (debugger, dict):
|
|
|
|
# This initializer is being run from LLDB in the embedded command interpreter
|
|
|
|
# Add any commands contained in this module to LLDB
|
|
|
|
debugger.HandleCommand('command script add -f monobt.monobt monobt')
|
2017-06-07 13:16:24 +00:00
|
|
|
debugger.HandleCommand('command alias mbt monobt')
|
2020-01-16 16:38:04 +00:00
|
|
|
print('"monobt" command installed')
|