| """Test stepping over vrs. hitting breakpoints & subsequent stepping in various forms.""" |
| |
| from lldbsuite.test.decorators import * |
| from lldbsuite.test.lldbtest import * |
| from lldbsuite.test import lldbutil |
| from lldbsuite.test_event.build_exception import BuildError |
| |
| |
| class StepUntilTestCase(TestBase): |
| def setUp(self): |
| # Call super's setUp(). |
| TestBase.setUp(self) |
| # Find the line numbers that we will step to in main: |
| self.main_source = "main.c" |
| self.less_than_two = line_number("main.c", "Less than 2") |
| self.greater_than_two = line_number("main.c", "Greater than or equal to 2.") |
| self.back_out_in_main = line_number("main.c", "Back out in main") |
| self.in_foo = line_number("main.c", "In foo") |
| |
| def _build_dict_for_discontinuity(self): |
| return dict( |
| CFLAGS_EXTRAS="-funique-basic-block-section-names " |
| + "-ffunction-sections -fbasic-block-sections=list=" |
| + self.getSourcePath("function.list"), |
| LD_EXTRAS="-Wl,--script=" + self.getSourcePath("symbol.order"), |
| ) |
| |
| def _common_setup(self, build_dict, args): |
| self.build(dictionary=build_dict) |
| exe = self.getBuildArtifact("a.out") |
| |
| target = self.dbg.CreateTarget(exe) |
| self.assertTrue(target, VALID_TARGET) |
| |
| main_source_spec = lldb.SBFileSpec(self.main_source) |
| break_before = target.BreakpointCreateBySourceRegex( |
| "At the start", main_source_spec |
| ) |
| self.assertTrue(break_before, VALID_BREAKPOINT) |
| |
| # Now launch the process, and do not stop at entry point. |
| process = target.LaunchSimple(args, None, self.get_process_working_directory()) |
| |
| self.assertTrue(process, PROCESS_IS_VALID) |
| |
| # The stop reason of the thread should be breakpoint. |
| threads = lldbutil.get_threads_stopped_at_breakpoint(process, break_before) |
| |
| if len(threads) != 1: |
| self.fail("Failed to stop at first breakpoint in main.") |
| |
| thread = threads[0] |
| return thread |
| |
| def do_until(self, args, until_lines, expected_linenum): |
| thread = self._common_setup(None, args) |
| |
| cmd_interp = self.dbg.GetCommandInterpreter() |
| ret_obj = lldb.SBCommandReturnObject() |
| |
| cmd_line = "thread until" |
| for line_num in until_lines: |
| cmd_line += " %d" % (line_num) |
| |
| cmd_interp.HandleCommand(cmd_line, ret_obj) |
| self.assertTrue( |
| ret_obj.Succeeded(), "'%s' failed: %s." % (cmd_line, ret_obj.GetError()) |
| ) |
| |
| frame = thread.frames[0] |
| line = frame.GetLineEntry().GetLine() |
| self.assertEqual( |
| line, expected_linenum, "Did not get the expected stop line number" |
| ) |
| |
| def test_hitting_one(self): |
| """Test thread step until - targeting one line and hitting it.""" |
| self.do_until(None, [self.less_than_two], self.less_than_two) |
| |
| def test_targetting_two_hitting_first(self): |
| """Test thread step until - targeting two lines and hitting one.""" |
| self.do_until( |
| ["foo", "bar", "baz"], |
| [self.less_than_two, self.greater_than_two], |
| self.greater_than_two, |
| ) |
| |
| def test_targetting_two_hitting_second(self): |
| """Test thread step until - targeting two lines and hitting the other one.""" |
| self.do_until( |
| None, [self.less_than_two, self.greater_than_two], self.less_than_two |
| ) |
| |
| def test_missing_one(self): |
| """Test thread step until - targeting one line and missing it by stepping out to call site""" |
| self.do_until( |
| ["foo", "bar", "baz"], [self.less_than_two], self.back_out_in_main |
| ) |
| |
| @no_debug_info_test |
| def test_bad_line(self): |
| """Test that we get an error if attempting to step outside the current |
| function""" |
| thread = self._common_setup(None, None) |
| self.expect( |
| f"thread until {self.in_foo}", |
| substrs=["Until target outside of the current function"], |
| error=True, |
| ) |
| |
| @no_debug_info_test |
| @skipIf(oslist=lldbplatformutil.getDarwinOSTriples() + ["windows"]) |
| @skipIf(archs=no_match(["x86_64", "aarch64"])) |
| @skipIf(compiler=no_match(["clang"])) |
| def test_bad_line_discontinuous(self): |
| """Test that we get an error if attempting to step outside the current |
| function -- and the function is discontinuous""" |
| try: |
| self.build(dictionary=self._build_dict_for_discontinuity()) |
| except BuildError as ex: |
| self.skipTest(f"failed to build with linker script.") |
| |
| _, _, thread, _ = lldbutil.run_to_source_breakpoint( |
| self, "At the start", lldb.SBFileSpec(self.main_source) |
| ) |
| self.expect( |
| f"thread until {self.in_foo}", |
| substrs=["Until target outside of the current function"], |
| error=True, |
| ) |