This patch improves breakpoint management when doing interactive scripted process debugging. In other to know which process set a breakpoint, we need to do some book keeping on the multiplexer scripted process. When initializing the multiplexer, we will first copy breakpoints that are already set on the driving target. Everytime we launch or resume, we should copy breakpoints from the multiplexer to the driving process. When creating a breakpoint from a child process, it needs to be set both on the multiplexer and on the driving process. We also tag the created breakpoint with the name and pid of the originator process. This patch also implements all the requirement to achieve proper breakpoint management. That involves: - Adding python interator for breakpoints and watchpoints in SBTarget - Add a new `ScriptedProcess.create_breakpoint` python method Differential Revision: https://reviews.llvm.org/D148548 Signed-off-by: Med Ismail Bennani <medismail.bennani@gmail.com>
123 lines
5.2 KiB
Python
123 lines
5.2 KiB
Python
"""
|
|
Test the functionality of interactive scripted processes
|
|
"""
|
|
|
|
import lldb
|
|
import lldbsuite.test.lldbutil as lldbutil
|
|
from lldbsuite.test.lldbtest import *
|
|
import json, os
|
|
|
|
|
|
class TestInteractiveScriptedProcess(TestBase):
|
|
|
|
NO_DEBUG_INFO_TESTCASE = True
|
|
|
|
def test_passthrough_launch(self):
|
|
"""Test a simple pass-through process launch"""
|
|
self.build()
|
|
self.runCmd("file " + self.getBuildArtifact("a.out"), CURRENT_EXECUTABLE_SET)
|
|
self.main_source_file = lldb.SBFileSpec("main.cpp")
|
|
self.script_module = "interactive_scripted_process"
|
|
self.script_file = self.script_module + ".py"
|
|
self.passthrough_launch()
|
|
|
|
def duplicate_target(self, driving_target):
|
|
exe = driving_target.executable.fullpath
|
|
triple = driving_target.triple
|
|
return self.dbg.CreateTargetWithFileAndTargetTriple(exe, triple)
|
|
|
|
def get_launch_info(self, class_name, script_dict):
|
|
structured_data = lldb.SBStructuredData()
|
|
structured_data.SetFromJSON(json.dumps(script_dict))
|
|
|
|
launch_info = lldb.SBLaunchInfo(None)
|
|
launch_info.SetProcessPluginName("ScriptedProcess")
|
|
launch_info.SetScriptedProcessClassName(class_name)
|
|
launch_info.SetScriptedProcessDictionary(structured_data)
|
|
return launch_info
|
|
|
|
def passthrough_launch(self):
|
|
"""Test that a simple passthrough wrapper functions correctly"""
|
|
# First build the real target:
|
|
self.assertEqual(self.dbg.GetNumTargets(), 1)
|
|
real_target_id = 0
|
|
real_target = self.dbg.GetTargetAtIndex(real_target_id)
|
|
lldbutil.run_break_set_by_source_regexp(self, "Break here")
|
|
self.assertEqual(real_target.GetNumBreakpoints(), 1)
|
|
|
|
# Now source in the scripted module:
|
|
script_path = os.path.join(self.getSourceDir(), self.script_file)
|
|
self.runCmd(f"command script import '{script_path}'")
|
|
|
|
mux_target = self.duplicate_target(real_target)
|
|
self.assertTrue(mux_target.IsValid(), "duplicate target succeeded")
|
|
|
|
mux_class = f"{self.script_module}.MultiplexerScriptedProcess"
|
|
script_dict = {"driving_target_idx": real_target_id}
|
|
mux_launch_info = self.get_launch_info(mux_class, script_dict)
|
|
mux_process_listener = lldb.SBListener(
|
|
"lldb.test.interactive-scripted-process.listener"
|
|
)
|
|
mux_launch_info.SetPassthroughListener(mux_process_listener)
|
|
|
|
self.dbg.SetAsync(True)
|
|
error = lldb.SBError()
|
|
mux_process = mux_target.Launch(mux_launch_info, error)
|
|
self.assertSuccess(error, "Launched multiplexer scripted process")
|
|
self.assertTrue(mux_process.IsValid(), "Got a valid process")
|
|
|
|
# Check that the mux process started running
|
|
event = lldbutil.fetch_next_event(
|
|
self, mux_process_listener, mux_process.GetBroadcaster(), timeout=60 * 5
|
|
)
|
|
self.assertState(lldb.SBProcess.GetStateFromEvent(event), lldb.eStateRunning)
|
|
# Check that the real process started running
|
|
event = lldbutil.fetch_next_event(
|
|
self, self.dbg.GetListener(), mux_process.GetBroadcaster()
|
|
)
|
|
self.assertState(lldb.SBProcess.GetStateFromEvent(event), lldb.eStateRunning)
|
|
|
|
# Check that the real process stopped
|
|
event = lldbutil.fetch_next_event(
|
|
self, self.dbg.GetListener(), mux_process.GetBroadcaster(), timeout=60 * 5
|
|
)
|
|
self.assertState(lldb.SBProcess.GetStateFromEvent(event), lldb.eStateStopped)
|
|
# Check that the mux process stopped
|
|
event = lldbutil.fetch_next_event(
|
|
self, mux_process_listener, mux_process.GetBroadcaster(), timeout=60 * 5
|
|
)
|
|
self.assertState(lldb.SBProcess.GetStateFromEvent(event), lldb.eStateStopped)
|
|
|
|
real_process = real_target.GetProcess()
|
|
self.assertTrue(real_process.IsValid(), "Got a valid process")
|
|
self.assertState(
|
|
real_process.GetState(), lldb.eStateStopped, "Process is stopped"
|
|
)
|
|
|
|
# This is a passthrough, so the two processes should have the same state:
|
|
# Check that we got the right threads:
|
|
self.assertEqual(
|
|
len(real_process.threads),
|
|
len(mux_process.threads),
|
|
"Same number of threads",
|
|
)
|
|
for id in range(len(real_process.threads)):
|
|
real_pc = real_process.threads[id].frame[0].pc
|
|
mux_pc = mux_process.threads[id].frame[0].pc
|
|
self.assertEqual(real_pc, mux_pc, f"PC's equal for {id}")
|
|
|
|
lldbutil.run_break_set_by_source_regexp(self, "also break here")
|
|
self.assertEqual(mux_target.GetNumBreakpoints(), 2)
|
|
error = mux_process.Continue()
|
|
self.assertSuccess(error, "Resuming multiplexer scripted process")
|
|
self.assertTrue(mux_process.IsValid(), "Got a valid process")
|
|
|
|
event = lldbutil.fetch_next_event(
|
|
self, mux_process_listener, mux_process.GetBroadcaster()
|
|
)
|
|
self.assertState(lldb.SBProcess.GetStateFromEvent(event), lldb.eStateRunning)
|
|
event = lldbutil.fetch_next_event(
|
|
self, mux_process_listener, mux_process.GetBroadcaster()
|
|
)
|
|
self.assertState(lldb.SBProcess.GetStateFromEvent(event), lldb.eStateStopped)
|