Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 1 | #!/usr/bin/python |
| 2 | |
| 3 | #---------------------------------------------------------------------- |
| 4 | # Be sure to add the python path that points to the LLDB shared library. |
| 5 | # On MacOSX csh, tcsh: |
| 6 | # setenv PYTHONPATH /Applications/Xcode.app/Contents/SharedFrameworks/LLDB.framework/Resources/Python |
| 7 | # On MacOSX sh, bash: |
| 8 | # export PYTHONPATH=/Applications/Xcode.app/Contents/SharedFrameworks/LLDB.framework/Resources/Python |
| 9 | #---------------------------------------------------------------------- |
| 10 | |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 11 | from __future__ import print_function |
| 12 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 13 | import optparse |
| 14 | import os |
| 15 | import platform |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 16 | import re |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 17 | import resource |
| 18 | import sys |
| 19 | import time |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 20 | import types |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 21 | |
Serge Guelton | 1a12dd7 | 2019-03-26 14:46:15 +0000 | [diff] [blame^] | 22 | if sys.version_info.major == 2: |
| 23 | import commands as subprocess |
| 24 | else: |
| 25 | import subprocess |
| 26 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 27 | #---------------------------------------------------------------------- |
| 28 | # Code that auto imports LLDB |
| 29 | #---------------------------------------------------------------------- |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 30 | try: |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 31 | # Just try for LLDB in case PYTHONPATH is already correctly setup |
| 32 | import lldb |
| 33 | except ImportError: |
| 34 | lldb_python_dirs = list() |
| 35 | # lldb is not in the PYTHONPATH, try some defaults for the current platform |
| 36 | platform_system = platform.system() |
| 37 | if platform_system == 'Darwin': |
| 38 | # On Darwin, try the currently selected Xcode directory |
Serge Guelton | 1a12dd7 | 2019-03-26 14:46:15 +0000 | [diff] [blame^] | 39 | xcode_dir = subprocess.getoutput("xcode-select --print-path") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 40 | if xcode_dir: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 41 | lldb_python_dirs.append( |
| 42 | os.path.realpath( |
| 43 | xcode_dir + |
| 44 | '/../SharedFrameworks/LLDB.framework/Resources/Python')) |
| 45 | lldb_python_dirs.append( |
| 46 | xcode_dir + '/Library/PrivateFrameworks/LLDB.framework/Resources/Python') |
| 47 | lldb_python_dirs.append( |
| 48 | '/System/Library/PrivateFrameworks/LLDB.framework/Resources/Python') |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 49 | success = False |
| 50 | for lldb_python_dir in lldb_python_dirs: |
| 51 | if os.path.exists(lldb_python_dir): |
| 52 | if not (sys.path.__contains__(lldb_python_dir)): |
| 53 | sys.path.append(lldb_python_dir) |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 54 | try: |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 55 | import lldb |
| 56 | except ImportError: |
| 57 | pass |
| 58 | else: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 59 | print('imported lldb from: "%s"' % (lldb_python_dir)) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 60 | success = True |
| 61 | break |
| 62 | if not success: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 63 | print("error: couldn't locate the 'lldb' module, please set PYTHONPATH correctly") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 64 | sys.exit(1) |
| 65 | |
| 66 | |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 67 | class Timer: |
| 68 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 69 | def __enter__(self): |
| 70 | self.start = time.clock() |
| 71 | return self |
| 72 | |
| 73 | def __exit__(self, *args): |
| 74 | self.end = time.clock() |
| 75 | self.interval = self.end - self.start |
| 76 | |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 77 | |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 78 | class Action(object): |
| 79 | """Class that encapsulates actions to take when a thread stops for a reason.""" |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 80 | |
| 81 | def __init__(self, callback=None, callback_owner=None): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 82 | self.callback = callback |
| 83 | self.callback_owner = callback_owner |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 84 | |
| 85 | def ThreadStopped(self, thread): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 86 | assert False, "performance.Action.ThreadStopped(self, thread) must be overridden in a subclass" |
| 87 | |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 88 | |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 89 | class PlanCompleteAction (Action): |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 90 | |
| 91 | def __init__(self, callback=None, callback_owner=None): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 92 | Action.__init__(self, callback, callback_owner) |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 93 | |
| 94 | def ThreadStopped(self, thread): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 95 | if thread.GetStopReason() == lldb.eStopReasonPlanComplete: |
| 96 | if self.callback: |
| 97 | if self.callback_owner: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 98 | self.callback(self.callback_owner, thread) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 99 | else: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 100 | self.callback(thread) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 101 | return True |
| 102 | return False |
| 103 | |
| 104 | |
| 105 | class BreakpointAction (Action): |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 106 | |
| 107 | def __init__( |
| 108 | self, |
| 109 | callback=None, |
| 110 | callback_owner=None, |
| 111 | name=None, |
| 112 | module=None, |
| 113 | file=None, |
| 114 | line=None, |
| 115 | breakpoint=None): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 116 | Action.__init__(self, callback, callback_owner) |
| 117 | self.modules = lldb.SBFileSpecList() |
| 118 | self.files = lldb.SBFileSpecList() |
| 119 | self.breakpoints = list() |
| 120 | # "module" can be a list or a string |
| 121 | if breakpoint: |
| 122 | self.breakpoints.append(breakpoint) |
| 123 | else: |
| 124 | if module: |
| 125 | if isinstance(module, types.ListType): |
| 126 | for module_path in module: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 127 | self.modules.Append( |
| 128 | lldb.SBFileSpec(module_path, False)) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 129 | elif isinstance(module, types.StringTypes): |
| 130 | self.modules.Append(lldb.SBFileSpec(module, False)) |
| 131 | if name: |
| 132 | # "file" can be a list or a string |
| 133 | if file: |
| 134 | if isinstance(file, types.ListType): |
| 135 | self.files = lldb.SBFileSpecList() |
| 136 | for f in file: |
| 137 | self.files.Append(lldb.SBFileSpec(f, False)) |
| 138 | elif isinstance(file, types.StringTypes): |
| 139 | self.files.Append(lldb.SBFileSpec(file, False)) |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 140 | self.breakpoints.append( |
| 141 | self.target.BreakpointCreateByName( |
| 142 | name, self.modules, self.files)) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 143 | elif file and line: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 144 | self.breakpoints.append( |
| 145 | self.target.BreakpointCreateByLocation( |
| 146 | file, line)) |
| 147 | |
| 148 | def ThreadStopped(self, thread): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 149 | if thread.GetStopReason() == lldb.eStopReasonBreakpoint: |
| 150 | for bp in self.breakpoints: |
| 151 | if bp.GetID() == thread.GetStopReasonDataAtIndex(0): |
| 152 | if self.callback: |
| 153 | if self.callback_owner: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 154 | self.callback(self.callback_owner, thread) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 155 | else: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 156 | self.callback(thread) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 157 | return True |
| 158 | return False |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 159 | |
| 160 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 161 | class TestCase: |
| 162 | """Class that aids in running performance tests.""" |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 163 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 164 | def __init__(self): |
| 165 | self.verbose = False |
| 166 | self.debugger = lldb.SBDebugger.Create() |
| 167 | self.target = None |
| 168 | self.process = None |
| 169 | self.thread = None |
| 170 | self.launch_info = None |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 171 | self.done = False |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 172 | self.listener = self.debugger.GetListener() |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 173 | self.user_actions = list() |
| 174 | self.builtin_actions = list() |
| 175 | self.bp_id_to_dict = dict() |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 176 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 177 | def Setup(self, args): |
| 178 | self.launch_info = lldb.SBLaunchInfo(args) |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 179 | |
| 180 | def Run(self, args): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 181 | assert False, "performance.TestCase.Run(self, args) must be subclassed" |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 182 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 183 | def Launch(self): |
| 184 | if self.target: |
| 185 | error = lldb.SBError() |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 186 | self.process = self.target.Launch(self.launch_info, error) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 187 | if not error.Success(): |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 188 | print("error: %s" % error.GetCString()) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 189 | if self.process: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 190 | self.process.GetBroadcaster().AddListener(self.listener, |
| 191 | lldb.SBProcess.eBroadcastBitStateChanged | lldb.SBProcess.eBroadcastBitInterrupt) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 192 | return True |
| 193 | return False |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 194 | |
| 195 | def WaitForNextProcessEvent(self): |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 196 | event = None |
| 197 | if self.process: |
| 198 | while event is None: |
| 199 | process_event = lldb.SBEvent() |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 200 | if self.listener.WaitForEvent(lldb.UINT32_MAX, process_event): |
| 201 | state = lldb.SBProcess.GetStateFromEvent(process_event) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 202 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 203 | print("event = %s" % (lldb.SBDebugger.StateAsCString(state))) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 204 | if lldb.SBProcess.GetRestartedFromEvent(process_event): |
| 205 | continue |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 206 | if state == lldb.eStateInvalid or state == lldb.eStateDetached or state == lldb.eStateCrashed or state == lldb.eStateUnloaded or state == lldb.eStateExited: |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 207 | event = process_event |
| 208 | self.done = True |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 209 | elif state == lldb.eStateConnected or state == lldb.eStateAttaching or state == lldb.eStateLaunching or state == lldb.eStateRunning or state == lldb.eStateStepping or state == lldb.eStateSuspended: |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 210 | continue |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 211 | elif state == lldb.eStateStopped: |
| 212 | event = process_event |
| 213 | call_test_step = True |
| 214 | fatal = False |
| 215 | selected_thread = False |
| 216 | for thread in self.process: |
| 217 | frame = thread.GetFrameAtIndex(0) |
| 218 | select_thread = False |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 219 | |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 220 | stop_reason = thread.GetStopReason() |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 221 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 222 | print("tid = %#x pc = %#x " % (thread.GetThreadID(), frame.GetPC()), end=' ') |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 223 | if stop_reason == lldb.eStopReasonNone: |
| 224 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 225 | print("none") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 226 | elif stop_reason == lldb.eStopReasonTrace: |
| 227 | select_thread = True |
| 228 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 229 | print("trace") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 230 | elif stop_reason == lldb.eStopReasonPlanComplete: |
| 231 | select_thread = True |
| 232 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 233 | print("plan complete") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 234 | elif stop_reason == lldb.eStopReasonThreadExiting: |
| 235 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 236 | print("thread exiting") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 237 | elif stop_reason == lldb.eStopReasonExec: |
| 238 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 239 | print("exec") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 240 | elif stop_reason == lldb.eStopReasonInvalid: |
| 241 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 242 | print("invalid") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 243 | elif stop_reason == lldb.eStopReasonException: |
| 244 | select_thread = True |
| 245 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 246 | print("exception") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 247 | fatal = True |
| 248 | elif stop_reason == lldb.eStopReasonBreakpoint: |
| 249 | select_thread = True |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 250 | bp_id = thread.GetStopReasonDataAtIndex(0) |
| 251 | bp_loc_id = thread.GetStopReasonDataAtIndex(1) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 252 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 253 | print("breakpoint id = %d.%d" % (bp_id, bp_loc_id)) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 254 | elif stop_reason == lldb.eStopReasonWatchpoint: |
| 255 | select_thread = True |
| 256 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 257 | print("watchpoint id = %d" % (thread.GetStopReasonDataAtIndex(0))) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 258 | elif stop_reason == lldb.eStopReasonSignal: |
| 259 | select_thread = True |
| 260 | if self.verbose: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 261 | print("signal %d" % (thread.GetStopReasonDataAtIndex(0))) |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 262 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 263 | if select_thread and not selected_thread: |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 264 | self.thread = thread |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 265 | selected_thread = self.process.SetSelectedThread( |
| 266 | thread) |
| 267 | |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 268 | for action in self.user_actions: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 269 | action.ThreadStopped(thread) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 270 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 271 | if fatal: |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 272 | # if self.verbose: |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 273 | # Xcode.RunCommand(self.debugger,"bt all",true) |
| 274 | sys.exit(1) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 275 | return event |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 276 | |
| 277 | |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 278 | class Measurement: |
| 279 | '''A class that encapsulates a measurement''' |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 280 | |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 281 | def __init__(self): |
| 282 | object.__init__(self) |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 283 | |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 284 | def Measure(self): |
| 285 | assert False, "performance.Measurement.Measure() must be subclassed" |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 286 | |
| 287 | |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 288 | class MemoryMeasurement(Measurement): |
| 289 | '''A class that can measure memory statistics for a process.''' |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 290 | |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 291 | def __init__(self, pid): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 292 | Measurement.__init__(self) |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 293 | self.pid = pid |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 294 | self.stats = [ |
| 295 | "rprvt", |
| 296 | "rshrd", |
| 297 | "rsize", |
| 298 | "vsize", |
| 299 | "vprvt", |
| 300 | "kprvt", |
| 301 | "kshrd", |
| 302 | "faults", |
| 303 | "cow", |
| 304 | "pageins"] |
| 305 | self.command = "top -l 1 -pid %u -stats %s" % ( |
| 306 | self.pid, ",".join(self.stats)) |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 307 | self.value = dict() |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 308 | |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 309 | def Measure(self): |
Serge Guelton | 1a12dd7 | 2019-03-26 14:46:15 +0000 | [diff] [blame^] | 310 | output = subprocess.getoutput(self.command).split("\n")[-1] |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 311 | values = re.split('[-+\s]+', output) |
| 312 | for (idx, stat) in enumerate(values): |
| 313 | multiplier = 1 |
| 314 | if stat: |
| 315 | if stat[-1] == 'K': |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 316 | multiplier = 1024 |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 317 | stat = stat[:-1] |
| 318 | elif stat[-1] == 'M': |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 319 | multiplier = 1024 * 1024 |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 320 | stat = stat[:-1] |
| 321 | elif stat[-1] == 'G': |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 322 | multiplier = 1024 * 1024 * 1024 |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 323 | elif stat[-1] == 'T': |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 324 | multiplier = 1024 * 1024 * 1024 * 1024 |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 325 | stat = stat[:-1] |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 326 | self.value[self.stats[idx]] = int(stat) * multiplier |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 327 | |
| 328 | def __str__(self): |
| 329 | '''Dump the MemoryMeasurement current value''' |
| 330 | s = '' |
| 331 | for key in self.value.keys(): |
| 332 | if s: |
| 333 | s += "\n" |
| 334 | s += "%8s = %s" % (key, self.value[key]) |
| 335 | return s |
| 336 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 337 | |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 338 | class TesterTestCase(TestCase): |
| 339 | |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 340 | def __init__(self): |
| 341 | TestCase.__init__(self) |
| 342 | self.verbose = True |
| 343 | self.num_steps = 5 |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 344 | |
| 345 | def BreakpointHit(self, thread): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 346 | bp_id = thread.GetStopReasonDataAtIndex(0) |
| 347 | loc_id = thread.GetStopReasonDataAtIndex(1) |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 348 | print("Breakpoint %i.%i hit: %s" % (bp_id, loc_id, thread.process.target.FindBreakpointByID(bp_id))) |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 349 | thread.StepOver() |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 350 | |
| 351 | def PlanComplete(self, thread): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 352 | if self.num_steps > 0: |
| 353 | thread.StepOver() |
| 354 | self.num_steps = self.num_steps - 1 |
| 355 | else: |
| 356 | thread.process.Kill() |
| 357 | |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 358 | def Run(self, args): |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 359 | self.Setup(args) |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 360 | with Timer() as total_time: |
| 361 | self.target = self.debugger.CreateTarget(args[0]) |
| 362 | if self.target: |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 363 | with Timer() as breakpoint_timer: |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 364 | bp = self.target.BreakpointCreateByName("main") |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 365 | print( |
| 366 | 'Breakpoint time = %.03f sec.' % |
| 367 | breakpoint_timer.interval) |
| 368 | |
| 369 | self.user_actions.append( |
| 370 | BreakpointAction( |
| 371 | breakpoint=bp, |
| 372 | callback=TesterTestCase.BreakpointHit, |
| 373 | callback_owner=self)) |
| 374 | self.user_actions.append( |
| 375 | PlanCompleteAction( |
| 376 | callback=TesterTestCase.PlanComplete, |
| 377 | callback_owner=self)) |
| 378 | |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 379 | if self.Launch(): |
Greg Clayton | 7b619fc | 2013-04-03 22:59:27 +0000 | [diff] [blame] | 380 | while not self.done: |
| 381 | self.WaitForNextProcessEvent() |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 382 | else: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 383 | print("error: failed to launch process") |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 384 | else: |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 385 | print("error: failed to create target with '%s'" % (args[0])) |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 386 | print('Total time = %.03f sec.' % total_time.interval) |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 387 | |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 388 | |
| 389 | if __name__ == '__main__': |
| 390 | lldb.SBDebugger.Initialize() |
| 391 | test = TesterTestCase() |
Kate Stone | b9c1b51 | 2016-09-06 20:57:50 +0000 | [diff] [blame] | 392 | test.Run(sys.argv[1:]) |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 393 | mem = MemoryMeasurement(os.getpid()) |
| 394 | mem.Measure() |
Serge Guelton | 525cd59 | 2019-03-21 18:27:40 +0000 | [diff] [blame] | 395 | print(str(mem)) |
Greg Clayton | 52e6378 | 2013-03-23 01:44:48 +0000 | [diff] [blame] | 396 | lldb.SBDebugger.Terminate() |
Greg Clayton | 2d95f35 | 2013-03-26 21:00:29 +0000 | [diff] [blame] | 397 | # print "sleeeping for 100 seconds" |
| 398 | # time.sleep(100) |