This patch introduces a new scripting affordance in lldb: `ScriptedFrame`. This allows user to produce mock stackframes in scripted threads and scripted processes from a python script. With this change, StackFrame can be synthetized from different sources: - Either from a dictionary containing a load address, and a frame index, which is the legacy way. - Or by creating a ScriptedFrame python object. One particularity of synthezising stackframes from the ScriptedFrame python object, is that these frame have an optional PC, meaning that they don't have a report a valid PC and they can act as shells that just contain static information, like the frame function name, the list of variables or registers, etc. It can also provide a symbol context. rdar://157260006 Signed-off-by: Med Ismail Bennani <ismail@bennani.ma> Signed-off-by: Med Ismail Bennani <ismail@bennani.ma>
195 lines
5.5 KiB
Python
195 lines
5.5 KiB
Python
import os, struct, signal
|
|
|
|
from typing import Any, Dict
|
|
|
|
import lldb
|
|
from lldb.plugins.scripted_process import ScriptedProcess
|
|
from lldb.plugins.scripted_process import ScriptedThread
|
|
from lldb.plugins.scripted_process import ScriptedFrame
|
|
|
|
|
|
class DummyStopHook:
|
|
def __init__(self, target, args):
|
|
self.target = target
|
|
self.args = args
|
|
|
|
def handle_stop(self, exe_ctx, stream):
|
|
print("My DummyStopHook triggered. Printing args: \n%s" % self.args)
|
|
sp = exe_ctx.process.GetScriptedImplementation()
|
|
sp.handled_stop = True
|
|
|
|
class DummyScriptedProcess(ScriptedProcess):
|
|
memory = None
|
|
|
|
def __init__(self, exe_ctx: lldb.SBExecutionContext, args: lldb.SBStructuredData):
|
|
super().__init__(exe_ctx, args)
|
|
self.threads[0] = DummyScriptedThread(self, args)
|
|
self.memory = {}
|
|
addr = 0x500000000
|
|
debugger = self.target.GetDebugger()
|
|
index = debugger.GetIndexOfTarget(self.target)
|
|
self.memory[addr] = "Hello, target " + str(index)
|
|
self.handled_stop = False
|
|
|
|
def read_memory_at_address(
|
|
self, addr: int, size: int, error: lldb.SBError
|
|
) -> lldb.SBData:
|
|
data = lldb.SBData().CreateDataFromCString(
|
|
self.target.GetByteOrder(), self.target.GetCodeByteSize(), self.memory[addr]
|
|
)
|
|
|
|
return data
|
|
|
|
def write_memory_at_address(self, addr, data, error):
|
|
self.memory[addr] = data.GetString(error, 0)
|
|
return len(self.memory[addr]) + 1
|
|
|
|
def get_loaded_images(self):
|
|
return self.loaded_images
|
|
|
|
def get_process_id(self) -> int:
|
|
return 42
|
|
|
|
def should_stop(self) -> bool:
|
|
return True
|
|
|
|
def is_alive(self) -> bool:
|
|
return True
|
|
|
|
def get_scripted_thread_plugin(self):
|
|
return DummyScriptedThread.__module__ + "." + DummyScriptedThread.__name__
|
|
|
|
def my_super_secret_method(self):
|
|
if hasattr(self, "my_super_secret_member"):
|
|
return self.my_super_secret_member
|
|
else:
|
|
return None
|
|
|
|
|
|
class DummyScriptedThread(ScriptedThread):
|
|
def __init__(self, process, args):
|
|
super().__init__(process, args)
|
|
self.frames.append({"pc": 0x0100001B00})
|
|
self.frames.append(DummyScriptedFrame(self, args, len(self.frames), "baz123"))
|
|
self.frames.append(DummyScriptedFrame(self, args, len(self.frames), "bar"))
|
|
self.frames.append(DummyScriptedFrame(self, args, len(self.frames), "foo"))
|
|
|
|
def get_thread_id(self) -> int:
|
|
return 0x19
|
|
|
|
def get_name(self) -> str:
|
|
return DummyScriptedThread.__name__ + ".thread-1"
|
|
|
|
def get_state(self) -> int:
|
|
return lldb.eStateStopped
|
|
|
|
def get_stop_reason(self) -> Dict[str, Any]:
|
|
return {"type": lldb.eStopReasonTrace, "data": {}}
|
|
|
|
def get_register_context(self) -> str:
|
|
return struct.pack(
|
|
"21Q",
|
|
1,
|
|
2,
|
|
3,
|
|
4,
|
|
5,
|
|
6,
|
|
7,
|
|
8,
|
|
9,
|
|
10,
|
|
11,
|
|
12,
|
|
13,
|
|
14,
|
|
15,
|
|
16,
|
|
17,
|
|
18,
|
|
19,
|
|
20,
|
|
21,
|
|
)
|
|
|
|
|
|
class DummyScriptedFrame(ScriptedFrame):
|
|
def __init__(self, thread, args, id, name, sym_ctx=None):
|
|
super().__init__(thread, args)
|
|
self.id = id
|
|
self.name = name
|
|
self.sym_ctx = sym_ctx
|
|
|
|
def get_id(self):
|
|
return self.id
|
|
|
|
def get_function_name(self):
|
|
return self.name
|
|
|
|
def get_register_context(self) -> str:
|
|
return struct.pack(
|
|
"21Q",
|
|
0x10001,
|
|
0x10002,
|
|
0x10003,
|
|
0x10004,
|
|
0x10005,
|
|
0x10006,
|
|
0x10007,
|
|
0x10008,
|
|
0x10009,
|
|
0x100010,
|
|
0x100011,
|
|
0x100012,
|
|
0x100013,
|
|
0x100014,
|
|
0x100015,
|
|
0x100016,
|
|
0x100017,
|
|
0x100018,
|
|
0x100019,
|
|
0x100020,
|
|
0x100021,
|
|
)
|
|
|
|
def get_symbol_context(self):
|
|
def get_symbol_context_for_function(func_name):
|
|
module = self.target.FindModule(self.target.GetExecutable())
|
|
if not module.IsValid():
|
|
return None
|
|
|
|
sym_ctx_list = module.FindFunctions(func_name)
|
|
if not sym_ctx_list.IsValid() or sym_ctx_list.GetSize() == 0:
|
|
return None
|
|
|
|
return sym_ctx_list.GetContextAtIndex(0)
|
|
|
|
return (
|
|
self.sym_ctx if self.sym_ctx else get_symbol_context_for_function(self.name)
|
|
)
|
|
|
|
def get_scripted_frame_plugin(self):
|
|
return DummyScriptedFrame.__module__ + "." + DummyScriptedFrame.__name__
|
|
|
|
|
|
def __lldb_init_module(debugger, dict):
|
|
# This is used when loading the script in an interactive debug session to
|
|
# automatically, register the stop-hook and launch the scripted process.
|
|
if not "SKIP_SCRIPTED_PROCESS_LAUNCH" in os.environ:
|
|
debugger.HandleCommand(
|
|
"target stop-hook add -k first -v 1 -k second -v 2 -P %s.%s"
|
|
% (__name__, DummyStopHook.__name__)
|
|
)
|
|
debugger.HandleCommand(
|
|
"process launch -C %s.%s" % (__name__, DummyScriptedProcess.__name__)
|
|
)
|
|
else:
|
|
print(
|
|
"Name of the class that will manage the scripted process: '%s.%s'"
|
|
% (__name__, DummyScriptedProcess.__name__)
|
|
)
|
|
print(
|
|
"Name of the class that will manage the stop-hook: '%s.%s'"
|
|
% (__name__, DummyStopHook.__name__)
|
|
)
|