2012-01-22 10:55:08 +08:00
|
|
|
#!/usr/bin/python
|
|
|
|
|
|
|
|
#----------------------------------------------------------------------
|
|
|
|
# Be sure to add the python path that points to the LLDB shared library.
|
|
|
|
#
|
2012-01-26 10:56:24 +08:00
|
|
|
# # To use this in the embedded python interpreter using "lldb" just
|
2016-09-07 04:57:50 +08:00
|
|
|
# import it with the full path using the "command script import"
|
2012-01-26 10:56:24 +08:00
|
|
|
# command
|
|
|
|
# (lldb) command script import /path/to/cmdtemplate.py
|
2012-01-22 10:55:08 +08:00
|
|
|
#----------------------------------------------------------------------
|
|
|
|
|
|
|
|
import lldb
|
|
|
|
import commands
|
|
|
|
import optparse
|
|
|
|
import shlex
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2013-05-09 09:32:24 +08:00
|
|
|
def create_framestats_options():
|
|
|
|
usage = "usage: %prog [options]"
|
2016-09-07 04:57:50 +08:00
|
|
|
description = '''This command is meant to be an example of how to make an LLDB command that
|
2013-05-09 09:32:24 +08:00
|
|
|
does something useful, follows best practices, and exploits the SB API.
|
|
|
|
Specifically, this command computes the aggregate and average size of the variables in the current frame
|
|
|
|
and allows you to tweak exactly which variables are to be accounted in the computation.
|
2012-09-13 13:35:34 +08:00
|
|
|
'''
|
2016-09-07 04:57:50 +08:00
|
|
|
parser = optparse.OptionParser(
|
|
|
|
description=description,
|
|
|
|
prog='framestats',
|
|
|
|
usage=usage)
|
|
|
|
parser.add_option(
|
|
|
|
'-i',
|
|
|
|
'--in-scope',
|
|
|
|
action='store_true',
|
|
|
|
dest='inscope',
|
|
|
|
help='in_scope_only = True',
|
|
|
|
default=False)
|
|
|
|
parser.add_option(
|
|
|
|
'-a',
|
|
|
|
'--arguments',
|
|
|
|
action='store_true',
|
|
|
|
dest='arguments',
|
|
|
|
help='arguments = True',
|
|
|
|
default=False)
|
|
|
|
parser.add_option(
|
|
|
|
'-l',
|
|
|
|
'--locals',
|
|
|
|
action='store_true',
|
|
|
|
dest='locals',
|
|
|
|
help='locals = True',
|
|
|
|
default=False)
|
|
|
|
parser.add_option(
|
|
|
|
'-s',
|
|
|
|
'--statics',
|
|
|
|
action='store_true',
|
|
|
|
dest='statics',
|
|
|
|
help='statics = True',
|
|
|
|
default=False)
|
2012-09-13 13:35:34 +08:00
|
|
|
return parser
|
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2013-05-09 09:32:24 +08:00
|
|
|
def the_framestats_command(debugger, command, result, dict):
|
2016-09-07 04:57:50 +08:00
|
|
|
# Use the Shell Lexer to properly parse up command options just like a
|
2012-09-13 13:35:34 +08:00
|
|
|
# shell would
|
|
|
|
command_args = shlex.split(command)
|
2013-05-09 09:32:24 +08:00
|
|
|
parser = create_framestats_options()
|
2012-01-22 10:55:08 +08:00
|
|
|
try:
|
|
|
|
(options, args) = parser.parse_args(command_args)
|
|
|
|
except:
|
2013-02-26 07:01:08 +08:00
|
|
|
# if you don't handle exceptions, passing an incorrect argument to the OptionParser will cause LLDB to exit
|
|
|
|
# (courtesy of OptParse dealing with argument errors by throwing SystemExit)
|
2016-09-07 04:57:50 +08:00
|
|
|
result.SetError("option parsing failed")
|
2013-07-12 05:49:38 +08:00
|
|
|
return
|
2016-09-07 04:57:50 +08:00
|
|
|
|
2013-05-09 09:32:24 +08:00
|
|
|
# in a command - the lldb.* convenience variables are not to be used
|
|
|
|
# and their values (if any) are undefined
|
|
|
|
# this is the best practice to access those objects from within a command
|
|
|
|
target = debugger.GetSelectedTarget()
|
|
|
|
process = target.GetProcess()
|
|
|
|
thread = process.GetSelectedThread()
|
|
|
|
frame = thread.GetSelectedFrame()
|
|
|
|
if not frame.IsValid():
|
|
|
|
return "no frame here"
|
|
|
|
# from now on, replace lldb.<thing>.whatever with <thing>.whatever
|
2016-09-07 04:57:50 +08:00
|
|
|
variables_list = frame.GetVariables(
|
|
|
|
options.arguments,
|
|
|
|
options.locals,
|
|
|
|
options.statics,
|
|
|
|
options.inscope)
|
2013-05-09 09:32:24 +08:00
|
|
|
variables_count = variables_list.GetSize()
|
|
|
|
if variables_count == 0:
|
2013-07-12 05:49:38 +08:00
|
|
|
print >> result, "no variables here"
|
2013-05-09 09:32:24 +08:00
|
|
|
return
|
|
|
|
total_size = 0
|
2016-09-07 04:57:50 +08:00
|
|
|
for i in range(0, variables_count):
|
2013-05-09 09:32:24 +08:00
|
|
|
variable = variables_list.GetValueAtIndex(i)
|
|
|
|
variable_type = variable.GetType()
|
|
|
|
total_size = total_size + variable_type.GetByteSize()
|
|
|
|
average_size = float(total_size) / variables_count
|
2016-09-07 04:57:50 +08:00
|
|
|
print >>result, "Your frame has %d variables. Their total size is %d bytes. The average size is %f bytes" % (
|
|
|
|
variables_count, total_size, average_size)
|
2013-07-12 05:49:38 +08:00
|
|
|
# not returning anything is akin to returning success
|
2012-01-22 10:55:08 +08:00
|
|
|
|
2016-09-07 04:57:50 +08:00
|
|
|
|
|
|
|
def __lldb_init_module(debugger, dict):
|
|
|
|
# This initializer is being run from LLDB in the embedded command interpreter
|
|
|
|
# Make the options so we can generate the help text for the new LLDB
|
2012-09-13 13:35:34 +08:00
|
|
|
# command line command prior to registering it with LLDB below
|
2013-05-09 09:32:24 +08:00
|
|
|
parser = create_framestats_options()
|
|
|
|
the_framestats_command.__doc__ = parser.format_help()
|
2012-01-22 10:55:08 +08:00
|
|
|
# Add any commands contained in this module to LLDB
|
2016-09-07 04:57:50 +08:00
|
|
|
debugger.HandleCommand(
|
|
|
|
'command script add -f cmdtemplate.the_framestats_command framestats')
|
2013-05-09 09:32:24 +08:00
|
|
|
print 'The "framestats" command has been installed, type "help framestats" or "framestats --help" for detailed help.'
|