2017-05-27 13:04:31 -04:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#coding: utf-8
|
|
|
|
#
|
|
|
|
# Usage: run `command script import -r misc/lldb_cruby.py` on LLDB
|
|
|
|
#
|
|
|
|
|
|
|
|
import lldb
|
|
|
|
import commands
|
|
|
|
import os
|
|
|
|
import shlex
|
|
|
|
|
2017-05-28 10:17:25 -04:00
|
|
|
def lldb_init(debugger):
|
|
|
|
target = debugger.GetSelectedTarget()
|
|
|
|
global SIZEOF_VALUE
|
|
|
|
SIZEOF_VALUE = target.FindFirstType("VALUE").GetByteSize()
|
|
|
|
g = globals()
|
|
|
|
for enum in target.FindFirstGlobalVariable('ruby_dummy_gdb_enums'):
|
|
|
|
enum = enum.GetType()
|
|
|
|
members = enum.GetEnumMembers()
|
|
|
|
for i in xrange(0, members.GetSize()):
|
|
|
|
member = members.GetTypeEnumMemberAtIndex(i)
|
|
|
|
name = member.GetName()
|
|
|
|
value = member.GetValueAsUnsigned()
|
|
|
|
g[name] = value
|
2017-05-27 13:04:31 -04:00
|
|
|
|
2017-05-28 10:17:25 -04:00
|
|
|
def fixnum_p(x):
|
|
|
|
return x & RUBY_FIXNUM_FLAG != 0
|
2017-05-27 13:04:31 -04:00
|
|
|
|
2017-05-28 10:17:25 -04:00
|
|
|
def flonum_p(x):
|
|
|
|
return (x&RUBY_FLONUM_MASK) == RUBY_FLONUM_FLAG
|
2017-05-27 13:04:31 -04:00
|
|
|
|
|
|
|
def lldb_rp(debugger, command, result, internal_dict):
|
|
|
|
target = debugger.GetSelectedTarget()
|
|
|
|
process = target.GetProcess()
|
|
|
|
thread = process.GetSelectedThread()
|
|
|
|
frame = thread.GetSelectedFrame()
|
2017-05-30 22:42:10 -04:00
|
|
|
if frame.IsValid():
|
|
|
|
val = frame.EvaluateExpression(command)
|
|
|
|
else:
|
|
|
|
val = target.EvaluateExpression(command)
|
|
|
|
error = val.GetError()
|
|
|
|
if error.Fail():
|
|
|
|
print >> result, error
|
|
|
|
return
|
2017-05-27 13:04:31 -04:00
|
|
|
num = val.GetValueAsSigned()
|
2017-05-28 10:17:25 -04:00
|
|
|
if num == RUBY_Qfalse:
|
|
|
|
print >> result, 'false'
|
|
|
|
elif num == RUBY_Qtrue:
|
|
|
|
print >> result, 'true'
|
|
|
|
elif num == RUBY_Qnil:
|
|
|
|
print >> result, 'nil'
|
|
|
|
elif num == RUBY_Qundef:
|
2017-05-30 22:42:10 -04:00
|
|
|
print >> result, 'undef'
|
2017-05-28 10:17:25 -04:00
|
|
|
elif fixnum_p(num):
|
|
|
|
print >> result, num >> 1
|
|
|
|
elif flonum_p(num):
|
|
|
|
debugger.HandleCommand("print rb_float_value(%0#x)" % val.GetValueAsUnsigned())
|
|
|
|
elif num & RUBY_IMMEDIATE_MASK:
|
|
|
|
print >> result, 'immediate(%x)' % num
|
2017-05-27 13:04:31 -04:00
|
|
|
else:
|
|
|
|
tRBasic = target.FindFirstType("struct RBasic").GetPointerType()
|
|
|
|
val = val.Cast(tRBasic)
|
|
|
|
flags = val.GetValueForExpressionPath("->flags").GetValueAsUnsigned()
|
2017-05-28 10:17:25 -04:00
|
|
|
if (flags & RUBY_FL_PROMOTED) == RUBY_FL_PROMOTED:
|
|
|
|
print >> result, "[PROMOTED] "
|
2017-05-27 13:04:31 -04:00
|
|
|
flType = flags & RUBY_T_MASK
|
2017-05-28 10:17:25 -04:00
|
|
|
if flType == RUBY_T_NONE:
|
|
|
|
print >> result, 'T_NONE: %s' % val.Dereference()
|
|
|
|
elif flType == RUBY_T_NIL:
|
|
|
|
print >> result, 'T_NIL: %s' % val.Dereference()
|
|
|
|
elif flType == RUBY_T_OBJECT:
|
|
|
|
tRObject = target.FindFirstType("struct RObject").GetPointerType()
|
|
|
|
val = val.Cast(tRObject)
|
|
|
|
print >> result, 'T_OBJECT: %s' % val.Dereference()
|
|
|
|
elif flType == RUBY_T_STRING:
|
2017-05-27 13:04:31 -04:00
|
|
|
tRString = target.FindFirstType("struct RString").GetPointerType()
|
|
|
|
val = val.Cast(tRString)
|
|
|
|
if flags & RSTRING_NOEMBED:
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, val.GetValueForExpressionPath("->as.heap")
|
2017-05-27 13:04:31 -04:00
|
|
|
else:
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, val.GetValueForExpressionPath("->as.ary")
|
2017-05-27 13:04:31 -04:00
|
|
|
elif flType == RUBY_T_ARRAY:
|
|
|
|
tRArray = target.FindFirstType("struct RArray").GetPointerType()
|
|
|
|
val = val.Cast(tRArray)
|
|
|
|
if flags & RUBY_FL_USER1:
|
|
|
|
len = ((flags & (RUBY_FL_USER3|RUBY_FL_USER4)) >> (RUBY_FL_USHIFT+3))
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, "T_ARRAY: len=%d (embed)" % len
|
2017-05-27 13:04:31 -04:00
|
|
|
if len == 0:
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, "{(empty)}"
|
2017-05-27 13:04:31 -04:00
|
|
|
else:
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, val.GetValueForExpressionPath("->as.ary")
|
2017-05-27 13:04:31 -04:00
|
|
|
else:
|
|
|
|
len = val.GetValueForExpressionPath("->as.heap.len").GetValueAsSigned()
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, "T_ARRAY: len=%d " % len
|
|
|
|
#print >> result, val.GetValueForExpressionPath("->as.heap")
|
2017-05-27 13:04:31 -04:00
|
|
|
if flags & RUBY_FL_USER2:
|
|
|
|
shared = val.GetValueForExpressionPath("->as.heap.aux.shared").GetValueAsUnsigned()
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, "(shared) shared=%016x " % shared
|
2017-05-27 13:04:31 -04:00
|
|
|
else:
|
|
|
|
capa = val.GetValueForExpressionPath("->as.heap.aux.capa").GetValueAsSigned()
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, "(ownership) capa=%d " % capa
|
2017-05-27 13:04:31 -04:00
|
|
|
if len == 0:
|
2017-05-28 10:17:25 -04:00
|
|
|
print >> result, "{(empty)}"
|
2017-05-27 13:04:31 -04:00
|
|
|
else:
|
2017-05-28 10:17:25 -04:00
|
|
|
debugger.HandleCommand("expression -Z %d -fx -- (const VALUE*)((struct RArray*)%d)->as.heap.ptr" % (len, val.GetValueAsUnsigned()))
|
2017-05-27 13:04:31 -04:00
|
|
|
debugger.HandleCommand("p (struct RArray *) %0#x" % val.GetValueAsUnsigned())
|
|
|
|
|
|
|
|
|
|
|
|
def __lldb_init_module(debugger, internal_dict):
|
|
|
|
debugger.HandleCommand("command script add -f lldb_cruby.lldb_rp rp")
|
2017-05-28 10:17:25 -04:00
|
|
|
lldb_init(debugger)
|
2017-05-27 13:04:31 -04:00
|
|
|
print "lldb scripts for ruby has been installed."
|