| """ Test that stop-on-sharedlibrary-events works and cooperates with breakpoints. """ |
| import lldb |
| from lldbsuite.test.decorators import * |
| from lldbsuite.test.lldbtest import * |
| from lldbsuite.test import lldbutil |
| |
| class TestStopOnSharedlibraryEvents(TestBase): |
| |
| mydir = TestBase.compute_mydir(__file__) |
| |
| @skipIfRemote |
| @skipIfWindows |
| @no_debug_info_test |
| def test_stopping_breakpoints(self): |
| self.do_test() |
| |
| @skipIfRemote |
| @skipIfWindows |
| @no_debug_info_test |
| def test_auto_continue(self): |
| def auto_continue(bkpt): |
| bkpt.SetAutoContinue(True) |
| self.do_test(auto_continue) |
| |
| @skipIfRemote |
| @skipIfWindows |
| @no_debug_info_test |
| def test_failing_condition(self): |
| def condition(bkpt): |
| bkpt.SetCondition("1 == 2") |
| self.do_test(condition) |
| |
| @skipIfRemote |
| @skipIfWindows |
| @no_debug_info_test |
| def test_continue_callback(self): |
| def bkpt_callback(bkpt): |
| bkpt.SetScriptCallbackBody("return False") |
| self.do_test(bkpt_callback) |
| |
| def do_test(self, bkpt_modifier = None): |
| self.build() |
| main_spec = lldb.SBFileSpec("main.cpp") |
| # Launch and stop before the dlopen call. |
| target, process, thread, _ = lldbutil.run_to_source_breakpoint(self, |
| "// Set a breakpoint here", main_spec, extra_images=["load_a", |
| "load_b"]) |
| |
| # Now turn on shared library events, continue and make sure we stop for the event. |
| self.runCmd("settings set target.process.stop-on-sharedlibrary-events 1") |
| self.addTearDownHook(lambda: self.runCmd( |
| "settings set target.process.stop-on-sharedlibrary-events 0")) |
| |
| # Since I don't know how to check that we are at the "right place" to stop for |
| # shared library events, make an breakpoint after the load is done and |
| # make sure we don't stop there: |
| backstop_bkpt_1 = target.BreakpointCreateBySourceRegex("Set another here - we should not hit this one", main_spec) |
| self.assertGreater(backstop_bkpt_1.GetNumLocations(), 0, "Set our second breakpoint") |
| |
| process.Continue() |
| self.assertEqual(process.GetState(), lldb.eStateStopped, "We didn't stop for the load") |
| self.assertEqual(backstop_bkpt_1.GetHitCount(), 0, "Hit our backstop breakpoint") |
| |
| # We should be stopped after the library is loaded, check that: |
| found_it = False |
| for module in target.modules: |
| if module.file.basename.find("load_a") > -1: |
| found_it = True |
| break |
| self.assertTrue(found_it, "Found the loaded module.") |
| |
| # Now capture the place where we stopped so we can set a breakpoint and make |
| # sure the breakpoint there works correctly: |
| load_address = process.GetSelectedThread().frames[0].addr |
| load_bkpt = target.BreakpointCreateBySBAddress(load_address) |
| self.assertGreater(load_bkpt.GetNumLocations(), 0, "Set the load breakpoint") |
| |
| backstop_bkpt_1.SetEnabled(False) |
| |
| backstop_bkpt_2 = target.BreakpointCreateBySourceRegex("Set a third here - we should not hit this one", main_spec) |
| self.assertGreater(backstop_bkpt_2.GetNumLocations(), 0, "Set our third breakpoint") |
| |
| if bkpt_modifier == None: |
| process.Continue() |
| self.assertEqual(process.GetState(), lldb.eStateStopped, "We didn't stop for the load") |
| self.assertEqual(backstop_bkpt_2.GetHitCount(), 0, "Hit our backstop breakpoint") |
| self.assertEqual(thread.stop_reason, lldb.eStopReasonBreakpoint, "We attributed the stop to the breakpoint") |
| self.assertEqual(load_bkpt.GetHitCount(), 1, "We hit our breakpoint at the load address") |
| else: |
| bkpt_modifier(load_bkpt) |
| process.Continue() |
| self.assertEqual(process.GetState(), lldb.eStateStopped, "We didn't stop") |
| self.assertTrue(thread.IsValid(), "Our thread was no longer valid.") |
| self.assertEqual(thread.stop_reason, lldb.eStopReasonBreakpoint, "We didn't hit some breakpoint") |
| self.assertEqual(backstop_bkpt_2.GetHitCount(), 1, "We continued to the right breakpoint") |
| |
| |
| |
| |
| |